【并发】【编程题】Lock 和 Condition

学习多线程有一段时间了,一直没有实践,从网上找了一道经典题来练练手。

题目

启动3个线程打印递增的数字,线程1先打印1,2,3,4,5,然后是线程2打印6,7,8,9,10,然后是线程3打印11,12,13,14 15。接着再由线程1打印16,17,18,19,20 ….以此类推,直到打印到75。程序的输出结果应该为:

线程1:1

线程1:2

线程1:3

线程1:4

线程1:5

线程2:6

线程2:7

线程2:8

线程2:9

线程2:10

……

解答一

该解法忽略了 synchronized 隐式锁只有一个任务等待队列,不能指定唤醒哪一个线程,因此会有不定的等待时间。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
package printnum;

import java.text.SimpleDateFormat;
import java.util.Date;

public class PrintNum {
private static final short MAX_NUM = 75;
private static final SimpleDateFormat DF = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
private volatile short num = 0;
private volatile boolean firstCondiction = true;
private volatile boolean secondCondiction = false;
private volatile boolean thirdCondiction = false;

private synchronized void printNum(short threadIndex) throws InterruptedException {
//循环执行,直到满足条件
while (num <= MAX_NUM) {
// synchronized 只有一个任务队列
// 唤醒所有线程,获取锁的不一定是期望的线程,有点类似"活锁",一直是其他两个线程获取到锁,但又不满足条件
// 因此等待到期望的线程需要运气
// 此处可能不阻塞,可能阻塞很久
if ((firstCondiction && threadIndex == 1)
|| (secondCondiction && threadIndex == 2)
|| (thirdCondiction && threadIndex == 3)) {
for (int i = 1; i <= 5; i++) {
num++;
if (num <= MAX_NUM) {
//特意加了时间来说明等待时间长短
System.out.println(DF.format(new Date()) + " 线程" + threadIndex + ":" + num);
}
}
// 按照1->2->3->1的顺序指定条件
switch (threadIndex) {
case 1:
firstCondiction = false;
secondCondiction = true;
thirdCondiction = false;
break;
case 2:
firstCondiction = false;
secondCondiction = false;
thirdCondiction = true;
break;
case 3:
firstCondiction = true;
thirdCondiction = false;
secondCondiction = false;
break;
default:
System.out.println("errorIndex:" + threadIndex);
break;
}
}
// 唤醒所有线程,线程重新获取锁
notifyAll();
//当前线程等待其他线程唤醒
wait(100);
}
wait(100);
}

private void printAllNum() {
//生成3个线程
for (short i = 1; i <= 3; i++) {
final short threadIndex = i;
Thread thread = new Thread(new Runnable() {
public void run() {
try {
printNum(threadIndex);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},
"线程" + i);
thread.start();
}
}

public static void main(String[] args) {
PrintNum printNum = new PrintNum();
printNum.printAllNum();
}
}

运行结果如下(注意数字20和21之间等待了42s):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
2019-09-22 17:36:19 线程1:1
2019-09-22 17:36:19 线程1:2
2019-09-22 17:36:19 线程1:3
2019-09-22 17:36:19 线程1:4
2019-09-22 17:36:19 线程1:5
2019-09-22 17:36:19 线程2:6
2019-09-22 17:36:19 线程2:7
2019-09-22 17:36:19 线程2:8
2019-09-22 17:36:19 线程2:9
2019-09-22 17:36:19 线程2:10
2019-09-22 17:36:19 线程3:11
2019-09-22 17:36:19 线程3:12
2019-09-22 17:36:19 线程3:13
2019-09-22 17:36:19 线程3:14
2019-09-22 17:36:19 线程3:15
2019-09-22 17:36:19 线程1:16
2019-09-22 17:36:19 线程1:17
2019-09-22 17:36:19 线程1:18
2019-09-22 17:36:19 线程1:19
2019-09-22 17:36:19 线程1:20
2019-09-22 17:37:01 线程2:21
2019-09-22 17:37:01 线程2:22
2019-09-22 17:37:01 线程2:23
2019-09-22 17:37:01 线程2:24
2019-09-22 17:37:01 线程2:25
2019-09-22 17:37:01 线程3:26
2019-09-22 17:37:01 线程3:27
2019-09-22 17:37:01 线程3:28
2019-09-22 17:37:01 线程3:29
2019-09-22 17:37:01 线程3:30
2019-09-22 17:37:01 线程1:31
2019-09-22 17:37:01 线程1:32
2019-09-22 17:37:01 线程1:33
2019-09-22 17:37:01 线程1:34
2019-09-22 17:37:01 线程1:35
2019-09-22 17:37:01 线程2:36
2019-09-22 17:37:01 线程2:37
2019-09-22 17:37:01 线程2:38
2019-09-22 17:37:01 线程2:39
2019-09-22 17:37:01 线程2:40
2019-09-22 17:37:01 线程3:41
2019-09-22 17:37:01 线程3:42
2019-09-22 17:37:01 线程3:43
2019-09-22 17:37:01 线程3:44
2019-09-22 17:37:01 线程3:45
2019-09-22 17:37:01 线程1:46
2019-09-22 17:37:01 线程1:47
2019-09-22 17:37:01 线程1:48
2019-09-22 17:37:01 线程1:49
2019-09-22 17:37:01 线程1:50
2019-09-22 17:37:01 线程2:51
2019-09-22 17:37:01 线程2:52
2019-09-22 17:37:01 线程2:53
2019-09-22 17:37:01 线程2:54
2019-09-22 17:37:01 线程2:55
2019-09-22 17:37:01 线程3:56
2019-09-22 17:37:01 线程3:57
2019-09-22 17:37:01 线程3:58
2019-09-22 17:37:01 线程3:59
2019-09-22 17:37:01 线程3:60
2019-09-22 17:37:01 线程1:61
2019-09-22 17:37:01 线程1:62
2019-09-22 17:37:01 线程1:63
2019-09-22 17:37:01 线程1:64
2019-09-22 17:37:01 线程1:65
2019-09-22 17:37:01 线程2:66
2019-09-22 17:37:01 线程2:67
2019-09-22 17:37:01 线程2:68
2019-09-22 17:37:01 线程2:69
2019-09-22 17:37:01 线程2:70
2019-09-22 17:37:01 线程3:71
2019-09-22 17:37:01 线程3:72
2019-09-22 17:37:01 线程3:73
2019-09-22 17:37:01 线程3:74
2019-09-22 17:37:01 线程3:75

解答二

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
package printnum;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class ConditionPrintNum {
private static final short MAX_NUM = 75;
private static final SimpleDateFormat DF = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
private final Lock lock = new ReentrantLock();
private final Condition condition1 = lock.newCondition();
private final Condition condition2 = lock.newCondition();
private final Condition condition3 = lock.newCondition();
private AtomicInteger num = new AtomicInteger(0);
private volatile short lastThreadIndex = 3;

//可以指定唤醒哪个队列的线程(signalCondition)
private void printNum(Condition currentCondition, Condition signalCondition, short threadIndex)
throws InterruptedException {
while (num.intValue() <= MAX_NUM) {
// 判断准入条件,直到满足
while ((lastThreadIndex == 3 && threadIndex != 1)
|| (lastThreadIndex == 1 && threadIndex != 2)
|| (lastThreadIndex == 2 && threadIndex != 3)) {
signalCondition.signalAll();
currentCondition.await(100, TimeUnit.MILLISECONDS);
}
for (int i = 1; i <= 5; i++) {
num.getAndAdd(1);
if (num.intValue() <= MAX_NUM) {
//特意加了时间来说明等待时间长短
System.out.println(DF.format(new Date()) + " 线程" + threadIndex + ":" + num);
}
}
lastThreadIndex = threadIndex;
signalCondition.signalAll();
currentCondition.await(100, TimeUnit.MILLISECONDS);
}
currentCondition.await(100, TimeUnit.MILLISECONDS);
}

private void printAllNum() {
Thread thread1 = new Thread(new Runnable() {
public void run() {
try {
lock.lock();
// 指定线程1完成后唤醒condition2线程,同时进入condition1排队
printNum(condition1, condition2, (short) 1);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
condition3.signal();
lock.unlock();
}
}
},
"线程1");
Thread thread2 = new Thread(new Runnable() {
public void run() {
try {
lock.lock();
// 指定线程2完成后唤醒condition3线程,同时进入condition2排队
printNum(condition2, condition3, (short) 2);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
condition1.signal();
lock.unlock();
}
}
},
"线程2");
Thread thread3 = new Thread(new Runnable() {
public void run() {
try {
lock.lock();
// 指定线程3完成后唤醒condition1线程,同时进入condition3排队
printNum(condition3, condition1, (short) 3);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
condition2.signal();
lock.unlock();
}
}
},
"线程3");
thread1.start();
thread2.start();
thread3.start();
}

public static void main(String[] args) {
ConditionPrintNum printNum = new ConditionPrintNum();
printNum.printAllNum();
}
}

运行效果如下(无特殊等待时间):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
2019-09-22 17:43:14 线程1:1
2019-09-22 17:43:14 线程1:2
2019-09-22 17:43:14 线程1:3
2019-09-22 17:43:14 线程1:4
2019-09-22 17:43:14 线程1:5
2019-09-22 17:43:14 线程2:6
2019-09-22 17:43:14 线程2:7
2019-09-22 17:43:14 线程2:8
2019-09-22 17:43:14 线程2:9
2019-09-22 17:43:14 线程2:10
2019-09-22 17:43:14 线程3:11
2019-09-22 17:43:14 线程3:12
2019-09-22 17:43:14 线程3:13
2019-09-22 17:43:14 线程3:14
2019-09-22 17:43:14 线程3:15
2019-09-22 17:43:14 线程1:16
2019-09-22 17:43:14 线程1:17
2019-09-22 17:43:14 线程1:18
2019-09-22 17:43:14 线程1:19
2019-09-22 17:43:14 线程1:20
2019-09-22 17:43:14 线程2:21
2019-09-22 17:43:14 线程2:22
2019-09-22 17:43:14 线程2:23
2019-09-22 17:43:14 线程2:24
2019-09-22 17:43:14 线程2:25
2019-09-22 17:43:14 线程3:26
2019-09-22 17:43:14 线程3:27
2019-09-22 17:43:14 线程3:28
2019-09-22 17:43:14 线程3:29
2019-09-22 17:43:14 线程3:30
2019-09-22 17:43:14 线程1:31
2019-09-22 17:43:14 线程1:32
2019-09-22 17:43:14 线程1:33
2019-09-22 17:43:14 线程1:34
2019-09-22 17:43:14 线程1:35
2019-09-22 17:43:14 线程2:36
2019-09-22 17:43:14 线程2:37
2019-09-22 17:43:14 线程2:38
2019-09-22 17:43:14 线程2:39
2019-09-22 17:43:14 线程2:40
2019-09-22 17:43:14 线程3:41
2019-09-22 17:43:14 线程3:42
2019-09-22 17:43:14 线程3:43
2019-09-22 17:43:14 线程3:44
2019-09-22 17:43:14 线程3:45
2019-09-22 17:43:14 线程1:46
2019-09-22 17:43:14 线程1:47
2019-09-22 17:43:14 线程1:48
2019-09-22 17:43:14 线程1:49
2019-09-22 17:43:14 线程1:50
2019-09-22 17:43:14 线程2:51
2019-09-22 17:43:14 线程2:52
2019-09-22 17:43:14 线程2:53
2019-09-22 17:43:14 线程2:54
2019-09-22 17:43:14 线程2:55
2019-09-22 17:43:14 线程3:56
2019-09-22 17:43:14 线程3:57
2019-09-22 17:43:14 线程3:58
2019-09-22 17:43:14 线程3:59
2019-09-22 17:43:14 线程3:60
2019-09-22 17:43:14 线程1:61
2019-09-22 17:43:14 线程1:62
2019-09-22 17:43:14 线程1:63
2019-09-22 17:43:14 线程1:64
2019-09-22 17:43:14 线程1:65
2019-09-22 17:43:14 线程2:66
2019-09-22 17:43:14 线程2:67
2019-09-22 17:43:14 线程2:68
2019-09-22 17:43:14 线程2:69
2019-09-22 17:43:14 线程2:70
2019-09-22 17:43:14 线程3:71
2019-09-22 17:43:14 线程3:72
2019-09-22 17:43:14 线程3:73
2019-09-22 17:43:14 线程3:74
2019-09-22 17:43:14 线程3:75
Newer Post

【并发】【编程题】Semaphore

信号量相关的编程题目,限流。 题目实现一个流控程序,控制客户端每秒调用某个远程服务不超过N次,客户端是会多线程并发调用。 解答123456789101112131415161718192021222324252627282930313233343536373839404142434445464748 …

Condition, Java, Lock, 多线程, 并发, 编程题 继续阅读
Older Post

【Java】技能框架&面试点

基础模块: 技术岗位与面试 计算机基础 JVM原理 多线程 设计模式 数据结构与算法 应用模块: 常用工具集 常用框架 缓存 队列 数据库 综合模块: 系统架构设计 微服务架构 容器化 …

Java, 框架, 面试 继续阅读